// 动态规划 - 核心 5 步：
// 1. 确定状态表示 - 根据 题目要求，经验(以 i,j 位置为结尾/开始......)，发现重复子问题 确定状态表示
// 2. 推导状态转移方程: dp[i] = ?
//    用 之前的状态 或者 之后的状态 推导当前的状态（根据最近一步划分问题）
// 3. 初始化：保证填表时不越界，结合多开数组的技巧
// 4. 确定填表顺序：填写当前状态值的时候，所需状态的值已经计算过了
// 5. 返回值：结合题目要求 + 状态表示

// 经典题目：斐波那契数列模型，路径问题，简单多状态

// 技巧：
// dp[] 表多开一个长度，处理数组越界及初始化复杂的问题
// dp[][] 表多开一行，多开一列
// 结合滚动数组优化 - 注意赋值顺序

// 总结经验:
// 动态规划题目如果定义完 dp[] 数组，发现 dp[i] 依赖前面的状态，也依赖后面的状态，那么想一想打家劫舍模型
// 如果觉得不像打家劫舍模型，那么搞一个数组预处理一下，搞成连续的数组，往打家劫舍模型上靠
// 如果题目的状态表示存在多个状态，比如给房子涂颜色（红蓝绿），某个位置元素（选或不选），
// 可以根据经验(以某个位置为结尾/开头)以及状态（定义多个状态: f[i], g[i]）定义状态表示
// 如果动态规划过程中涉及到状态转换，需要画状态机图进行分析

// 例题 6:
// 给定一个整数数组 prices，其中 prices[i]表示第 i 天的股票价格 ；整数 fee 代表了交易股票的手续费用。
// 你可以无限次地完成交易，但是你每笔交易都需要付手续费。如果你已经购买了一个股票，在卖出它之前你就不能再继续购买股票了。
// 回获得利润的最大值。
//
//        注意：这里的一笔交易指买入持有并卖出股票的整个过程，每笔交易你只需要为支付一次手续费。
//
//        示例 1：
//
//        输入：prices = [1, 3, 2, 8, 4, 9], fee = 2
//        输出：8
//        解释：能够达到的最大利润:
//        在此处买入 prices[0] = 1
//        在此处卖出 prices[3] = 8
//        在此处买入 prices[4] = 4
//        在此处卖出 prices[5] = 9
//        总利润: ((8 - 1) - 2) + ((9 - 4) - 2) = 8
//        示例 2：
//
//        输入：prices = [1,3,7,5,10,3], fee = 3
//        输出：6
//
//
//        提示：
//
//        1 <= prices.length <= 5 * 104
//        1 <= prices[i] < 5 * 104
//        0 <= fee < 5 * 104

// 解题思路:
// dp[i][0], dp[i][1] 表示第 i 天处于买入状态或者卖出状态的最大利润
// 第 i 天处于买入状态能获得最大利润是前一天处于买入状态或者卖出状态进行买入操作（含手续费）的最大值
// dp[i][0] = max(dp[i - 1][0], dp[i - 1][1] - price[i] - fee)
// 第 i 天处于卖出状态能获得的最大利润是前一天处于买入状态进行卖出操作或者处于卖出状态的最大值
// dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] + price)

public class MaxProfit2 {
    public int maxProfit(int[] prices, int fee) {
        int n = prices.length;
        int[][] dp = new int[n][2];

        dp[0][0] = -prices[0] - fee;
        dp[0][1] = 0;

        for(int i = 1; i < n; i++){
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] - prices[i] - fee);
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] + prices[i]);
        }

        return dp[n - 1][1];
    }
}
